import {
  ChangeDetectionStrategy,
  Component,
  computed,
  inject,
  input,
  signal,
} from '@angular/core';
import { toObservable, toSignal, takeUntilDestroyed } from '@angular/core/rxjs-interop';
import { FormsModule } from '@angular/forms';
import { CommonModule } from '@angular/common';
import { MatFormFieldModule } from '@angular/material/form-field';
import { MatInputModule } from '@angular/material/input';
import { CdkTextareaAutosize } from '@angular/cdk/text-field';
import { MatButtonModule } from '@angular/material/button';
import { MatIconModule } from '@angular/material/icon';
import { MatTooltipModule } from '@angular/material/tooltip';
import { TranslatePipe, TranslateService } from '@ngx-translate/core';
import { Subject } from 'rxjs';
import { debounceTime, distinctUntilChanged, switchMap, take } from 'rxjs/operators';
import { Metric, MetricCopy } from '../metric.model';
import { MetricService } from '../metric.service';
import { GlobalTrackingIntervalService } from '../../../core/global-tracking-interval/global-tracking-interval.service';
import { MatDialog } from '@angular/material/dialog';
import {
  ReflectionHistoryDialogComponent,
  ReflectionHistoryItem,
} from './reflection-history-dialog.component';
import { DEFAULT_METRIC_FOR_DAY } from '../metric.const';
import { T } from '../../../t.const';

@Component({
  selector: 'reflection-note',
  templateUrl: './reflection-note.component.html',
  styleUrls: ['./reflection-note.component.scss'],
  standalone: true,
  changeDetection: ChangeDetectionStrategy.OnPush,
  imports: [
    CommonModule,
    FormsModule,
    MatFormFieldModule,
    MatInputModule,
    CdkTextareaAutosize,
    MatButtonModule,
    MatIconModule,
    MatTooltipModule,
    TranslatePipe,
  ],
})
export class ReflectionNoteComponent {
  private readonly _metricService = inject(MetricService);
  private readonly _globalTrackingIntervalService = inject(GlobalTrackingIntervalService);
  private readonly _translate = inject(TranslateService);
  private readonly _matDialog = inject(MatDialog);

  readonly dayStr = input<string | null>(null);
  readonly T = T;

  private readonly _placeholderKeys: string[] = [
    T.F.METRIC.REFLECTION.PLACEHOLDER_1,
    T.F.METRIC.REFLECTION.PLACEHOLDER_2,
    T.F.METRIC.REFLECTION.PLACEHOLDER_3,
    T.F.METRIC.REFLECTION.PLACEHOLDER_4,
    T.F.METRIC.REFLECTION.PLACEHOLDER_5,
  ];
  readonly placeholder = signal<string>(this._pickRandomPlaceholder());

  private readonly _resolvedDay = computed(
    () => this.dayStr() ?? this._globalTrackingIntervalService.todayDateStr(),
  );

  private readonly _day$ = toObservable(this._resolvedDay).pipe(distinctUntilChanged());
  private readonly _metricForDay = toSignal(
    this._day$.pipe(switchMap((day) => this._metricService.getMetricForDay$(day))),
    { initialValue: { id: '', ...DEFAULT_METRIC_FOR_DAY } as MetricCopy },
  );

  readonly reflectionText = computed(() => {
    return this._metricForDay()?.reflections?.[0]?.text ?? '';
  });

  private readonly _reflectionChanges$ = new Subject<string>();

  constructor() {
    this._reflectionChanges$
      .pipe(debounceTime(500), takeUntilDestroyed())
      .subscribe((value) => {
        this._persistReflection(value);
      });

    this._translate.onLangChange
      .pipe(takeUntilDestroyed())
      .subscribe(() => this.placeholder.set(this._pickRandomPlaceholder()));
  }

  onReflectionChange(value: string): void {
    this._reflectionChanges$.next(value ?? '');
  }

  async openHistory(): Promise<void> {
    const metrics =
      (await this._metricService.getAllMetrics$().pipe(take(1)).toPromise()) ?? [];
    const entries = metrics
      .map((metric) => {
        const reflection = metric.reflections?.[0];
        const text = reflection?.text?.trim();
        if (!text) {
          return null;
        }
        return {
          day: metric.id,
          text,
          created: reflection?.created ?? this._dayToTimestamp(metric.id),
        } as ReflectionHistoryItem;
      })
      .filter((entry): entry is ReflectionHistoryItem => !!entry);

    const sortedEntries = entries.sort((a, b) => b.created - a.created);

    this._matDialog.open(ReflectionHistoryDialogComponent, {
      data: { entries: sortedEntries },
      width: '520px',
    });
  }

  private _persistReflection(value: string): void {
    const metric = this._metricForDay();
    if (!metric) {
      return;
    }
    const trimmed = value.trim();
    const existing = metric.reflections?.[0];
    const reflections =
      trimmed.length > 0
        ? [
            {
              text: trimmed,
              created: existing?.created ?? Date.now(),
            },
          ]
        : [];
    this._metricService.upsertMetric({
      ...metric,
      reflections,
    } as Metric);
  }

  private _dayToTimestamp(dayStr: string): number {
    return this._createLocalDateFromDay(dayStr).getTime();
  }

  private _createLocalDateFromDay(dayStr: string): Date {
    const [year, month, day] = dayStr
      .split('-')
      .map((value) => Number.parseInt(value, 10));
    return new Date(year, (month || 1) - 1, day || 1);
  }

  private _pickRandomPlaceholder(): string {
    const key =
      this._placeholderKeys[
        Math.floor(Math.random() * Math.max(this._placeholderKeys.length, 1))
      ];
    return this._translate.instant(key);
  }
}
